热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

表里|也就是_UE4Unlua源码解析7Lua通过UE命名空间访问C++类型的实现原理

篇首语:本文由编程笔记#小编为大家整理,主要介绍了UE4Unlua源码解析7-Lua通过UE命名空间访问C++类型的实现原理相关的知识,希望对你有一定的参考价值。

篇首语:本文由编程笔记#小编为大家整理,主要介绍了UE4 Unlua源码解析7 - Lua通过UE命名空间访问C++类型的实现原理相关的知识,希望对你有一定的参考价值。



Lua通过UE命名空间访问C++类型的实现原理


          • 1.1 UE4发生了 什么
          • 1.2 UE4.UKismetSystemLibrary发生了什么
          • 1.3 UE4.UKismetSystemLibrary.PrintString发生了什么
          • 1.4 UE4.UKismetSystemLibrary.PrintString(“Hello”)发生了什么






举例 UE4.UKismetSystemLibrary.PrintString(“hello”)

我们来看Unlua提供的例子的HelloWorld

代码是

local hello = “HelloWorld”
UE4.UKismetSystemLibrary.PrintString(hello)

这个例子我们最终会调到UE4的方法,并且成功执行,那么首先,我们要解答的问题就是为什么Lua这么写最终能调用到C++的方法,

而且Lua传的参数是Lua的字符串,为什么C++能成功执行,这简单的代码,背后到底做了什么事情


1.1 UE4发生了 什么

Unlua.lua中声明了UE4,可以看到UE4其实就是全局表_G,而且他的元表是global_mt,Index元方法为global_index,我们去看看global_index

第73行拿到传入的索引k,检查第一个字母是不是UE的类的首字母,是就调用RegisterClass,如果是E,就调用RegisterEnum,然后调用rawget,rawget方法是不使用Index元方法的取值函数,可以猜测上面的函数执行完之后,t表也就是UE4表里已经就有了索引k的值了。

具体的执行逻辑接着看


1.2 UE4.UKismetSystemLibrary发生了什么

由上节讲到的,UE4.UKismetSystemLibrary的时候,因为UE4是表,所以就去表里找UKismetSystemLibrary,但是第一次肯定是找不到的,于是走到UE4的元表的元方法Index,也就是执行global_index,传进来的t是UE4,k是“UKismetSystemLibrary”

由之前的分析可以看到代码会走到RegisterClass,然后传入k

RegisterClass在lua测的声明如下

看到LuaContext.cpp里

lua api lua_register前面讲到了,将C函数f设置为全局名称name的新值,lua端可以通过name调用C方法f

所以这行代码将C函数Global_RegisterClass注册,在Lua端通过Lua全局名称RegisterClass调用

Global_RegisterClass和RegisterClass逐行解释之前讲过了,此处只说结果

Global_RegisterClass 读虚拟栈的大小,当小于1的时候返回,否则调用RegisterClass,参数传(虚拟机,栈底的值转成C 的string)



此时虚拟栈的内容是传进来类名“UKismetSystemLibrary”,以及参数的数量1,所以传递给RegisterClass的参数是(虚拟机L,“UKismetSystemLibrary”)


RegisterClass逐行解释之前讲过了,总之是两大部分

1 根据UE4反射生成UClass的描述信息FClassDesc ,然后存储到UnLua的存储全局反射数据的GReflectionRegistry对象中。

2 根据生成的FClassDesc 为其在Lua中注册table,然后设置table的元表为自己,然后往table里塞一些元方法

RegisterClass结束后,UE4.UKismetSystemLibrary就是一个Lua端的表了,这个表的元表是自己,并且元方法Index是LuaCore中的方法Class_Index

类图如下


1.3 UE4.UKismetSystemLibrary.PrintString发生了什么

截至目前,UE4.UKismetSystemLibrary就能拿到上一步生成的Lua table了,此时调用.PrintString的时候,可以看到表里没用这个数据,所以走到元表的Index元方法,之前我们知道,元方法为LuaCore中的Class_Index,所以重点看这个方法即可

其中最重要的是走到GetField

GetField和RegiestClass很像,就像是注册类一样,GetField主要也做两件事

1

​ 如果传进来的是属性,就根据UE4反射生成FPropertyDesc

​ 如果传进来的是方法,就根据UE4反射生成FFunctionDesc,并且把FPropertyDesc或FFunctionDesc存在FClassDesc中

2

​ 如果是属性 就在表里存FPropertyDesc,然后将FPropertyDesc压栈,作为返回值返回

​ 如果是方法,就在表里存FFunctionDesc+C方法Class_CallUFunction的闭包进去,FFunctionDesc作为Upvalue。然后将FFunctionDesc+C方法Class_CallUFunction的闭包压栈,作为返回值返回

此时的类图如下


1.4 UE4.UKismetSystemLibrary.PrintString(“Hello”)发生了什么

截至目前,相当于调用LuaCore的Class_CallUFunction

传进来的参数是lua的string hello

根据之前的解释C++通过闭包里的FFunctionDesc执行方法CallUE,参数是Lua传进去的,通过

1)执行PreCall,根据【UFunctionDesc】和Lua参数,将Lua参数转换成C++参数,再根据【UFunctionDesc】反射信息,将C++参数写入到一个缓存区中

2)执行 UObject::ProcessEvent(FinalFunction, Params),参数FinalFunction是UFunction,参数Params是前面保存有C++参数的缓存区。ProcessEvent执行时,即调用了真正我们想调用的C++ Create函数,然后把C++返回值放入了Params缓存区中

3)执行PostCall,从Params缓存区中读出C++返回值,转换成Lua返回值,Push进Lua栈,返回给Lua。

结束,至此,屏幕应该输出Hello


推荐阅读
  • 本文介绍了南邮ctf-web的writeup,包括签到题和md5 collision。在CTF比赛和渗透测试中,可以通过查看源代码、代码注释、页面隐藏元素、超链接和HTTP响应头部来寻找flag或提示信息。利用PHP弱类型,可以发现md5('QNKCDZO')='0e830400451993494058024219903391'和md5('240610708')='0e462097431906509019562988736854'。 ... [详细]
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • PHP图片截取方法及应用实例
    本文介绍了使用PHP动态切割JPEG图片的方法,并提供了应用实例,包括截取视频图、提取文章内容中的图片地址、裁切图片等问题。详细介绍了相关的PHP函数和参数的使用,以及图片切割的具体步骤。同时,还提供了一些注意事项和优化建议。通过本文的学习,读者可以掌握PHP图片截取的技巧,实现自己的需求。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • springmvc学习笔记(十):控制器业务方法中通过注解实现封装Javabean接收表单提交的数据
    本文介绍了在springmvc学习笔记系列的第十篇中,控制器的业务方法中如何通过注解实现封装Javabean来接收表单提交的数据。同时还讨论了当有多个注册表单且字段完全相同时,如何将其交给同一个控制器处理。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • 本文讨论了在Windows 8上安装gvim中插件时出现的错误加载问题。作者将EasyMotion插件放在了正确的位置,但加载时却出现了错误。作者提供了下载链接和之前放置插件的位置,并列出了出现的错误信息。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
author-avatar
鄙人fisher_779
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有